home *** CD-ROM | disk | FTP | other *** search
- #include "global.h"
- #include "mbuf.h"
- #include "ip.h"
- #include "usock.h"
- #include "socket.h"
-
- char Inet_eol[] = "\r\n";
-
- static void rip_recv __ARGS((struct raw_ip *rp));
- static void autobind __ARGS((struct usock *up));
-
- int
- so_ip_sock(up,protocol)
- struct usock *up;
- int protocol;
- {
- int s;
-
- s = up->index;
- up->cb.rip = raw_ip(protocol,rip_recv);
- up->cb.rip->user = s;
- return 0;
- }
- int
- so_ip_conn(up)
- struct usock *up;
- {
- if(up->name == NULLCHAR)
- autobind(up);
- return 0;
- }
- int
- so_ip_recv(up,bpp,from,fromlen)
- struct usock *up;
- struct mbuf **bpp;
- char *from;
- int *fromlen;
- {
- struct raw_ip *rip;
- struct sockaddr_in *remote;
- struct ip ip;
- int cnt;
-
- while((rip = up->cb.rip) != NULLRIP && rip->rcvq == NULLBUF){
- if(up->noblock){
- errno = EWOULDBLOCK;
- return -1;
- } else if((errno = pwait(up)) != 0){
- return -1;
- }
- }
- if(rip == NULLRIP){
- /* Connection went away */
- errno = ENOTCONN;
- return -1;
- }
- *bpp = dequeue(&rip->rcvq);
- ntohip(&ip,bpp);
-
- cnt = len_p(*bpp);
- if(from != NULLCHAR && fromlen != (int *)NULL && *fromlen >= SOCKSIZE){
- remote = (struct sockaddr_in *)from;
- remote->sin_family = AF_INET;
- remote->sin_addr.s_addr = ip.source;
- remote->sin_port = 0;
- *fromlen = SOCKSIZE;
- }
- return cnt;
- }
- int
- so_ip_send(up,bp,to)
- struct usock *up;
- struct mbuf *bp;
- char *to;
- {
- struct sockaddr_in *local,*remote;
-
- if(up->name == NULLCHAR)
- autobind(up);
- local = (struct sockaddr_in *)up->name;
- if(to != NULLCHAR){
- remote = (struct sockaddr_in *)to;
- } else if(up->peername != NULLCHAR) {
- remote = (struct sockaddr_in *)up->peername;
- } else {
- free_p(bp);
- errno = ENOTCONN;
- return -1;
- }
- ip_send(local->sin_addr.s_addr,remote->sin_addr.s_addr,
- (char)up->cb.rip->protocol,0,0,bp,0,0,0);
- return 0;
- }
- int
- so_ip_qlen(up,rtx)
- struct usock *up;
- int rtx;
- {
- int len;
-
- switch(rtx){
- case 0:
- len = len_q(up->cb.rip->rcvq);
- break;
- case 1:
- len = 0;
- break;
- }
- return len;
- }
- int
- so_ip_close(up)
- struct usock *up;
- {
- del_ip(up->cb.rip);
- return 0;
- }
- int
- checkipaddr(name,namelen)
- char *name;
- int namelen;
- {
- struct sockaddr_in *sock;
-
- sock = (struct sockaddr_in *)name;
- if(sock->sin_family != AF_INET || namelen != sizeof(struct sockaddr_in))
- return -1;
- return 0;
- }
-
- /* Raw IP receive upcall routine */
- static void
- rip_recv(rp)
- struct raw_ip *rp;
- {
- psignal(itop(rp->user),1);
- pwait(NULL);
- }
- /* Issue an automatic bind of a local address */
- static void
- autobind(up)
- struct usock *up;
- {
- struct sockaddr_in local;
- int s;
-
- s = up->index;
- local.sin_family = AF_INET;
- local.sin_addr.s_addr = INADDR_ANY;
- local.sin_port = Lport++;
- bind(s,(char *)&local,sizeof(struct sockaddr_in));
- }
- char *
- ippsocket(p)
- struct sockaddr *p;
- {
- struct sockaddr_in *sp;
- struct socket socket;
- static char buf[30];
-
- sp = (struct sockaddr_in *)p;
- socket.address = sp->sin_addr.s_addr;
- socket.port = sp->sin_port;
- strcpy(buf,pinet(&socket));
-
- return buf;
- }
-